Home | History | Annotate | Download | only in remote-processor
      1 /*
      2  * Copyright (c) 2011-2014, Intel Corporation
      3  * All rights reserved.
      4  *
      5  * Redistribution and use in source and binary forms, with or without modification,
      6  * are permitted provided that the following conditions are met:
      7  *
      8  * 1. Redistributions of source code must retain the above copyright notice, this
      9  * list of conditions and the following disclaimer.
     10  *
     11  * 2. Redistributions in binary form must reproduce the above copyright notice,
     12  * this list of conditions and the following disclaimer in the documentation and/or
     13  * other materials provided with the distribution.
     14  *
     15  * 3. Neither the name of the copyright holder nor the names of its contributors
     16  * may be used to endorse or promote products derived from this software without
     17  * specific prior written permission.
     18  *
     19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
     20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
     21  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
     22  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
     23  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
     24  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
     25  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
     26  * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
     28  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29  */
     30 #pragma once
     31 
     32 #include <vector>
     33 #include "RemoteCommandHandler.h"
     34 
     35 template <class CCommandParser>
     36 class TRemoteCommandHandlerTemplate : public IRemoteCommandHandler
     37 {
     38 public:
     39     /** Remote command parser execution return status */
     40     enum CommandStatus
     41     {
     42         EDone,      /**< Command succeded, return "Done" */
     43         ESucceeded, /**< Command succeeded */
     44         EFailed,    /**< Command failed */
     45         EShowUsage  /**< Command failed, show usage */
     46     };
     47 
     48     /** Type of the remote command callbacks
     49      *
     50      * @param[in] remoteCommand contains the arguments of the received command.
     51      * @param[out] strResult a string containing the result of the command.
     52      *
     53      * @return the command execution status, @see CommandStatus
     54      */
     55     typedef CommandStatus (CCommandParser::*RemoteCommandParser)(
     56         const IRemoteCommand &remoteCommand, std::string &strResult);
     57 
     58 private:
     59     // Parser descriptions
     60     class CRemoteCommandParserItem
     61     {
     62     public:
     63         CRemoteCommandParserItem(const std::string &strCommandName, RemoteCommandParser pfnParser,
     64                                  size_t minArgumentCount, const std::string &strHelp,
     65                                  const std::string &strDescription)
     66             : _strCommandName(strCommandName), _pfnParser(pfnParser),
     67               _minArgumentCount(minArgumentCount), _strHelp(strHelp),
     68               _strDescription(strDescription)
     69         {
     70         }
     71 
     72         const std::string &getCommandName() const { return _strCommandName; }
     73 
     74         const std::string &getDescription() const { return _strDescription; }
     75 
     76         // Usage
     77         std::string usage() const { return _strCommandName + " " + _strHelp; }
     78 
     79         bool parse(CCommandParser *pCommandParser, const IRemoteCommand &remoteCommand,
     80                    std::string &strResult) const
     81         {
     82             // Check enough arguments supplied
     83             if (remoteCommand.getArgumentCount() < _minArgumentCount) {
     84 
     85                 strResult = std::string("Not enough arguments supplied\nUsage:\n") + usage();
     86 
     87                 return false;
     88             }
     89 
     90             switch ((pCommandParser->*_pfnParser)(remoteCommand, strResult)) {
     91             case EDone:
     92                 strResult = "Done";
     93             // Fall through intentionally
     94             case ESucceeded:
     95                 return true;
     96             case EShowUsage:
     97                 strResult = usage();
     98             // Fall through intentionally
     99             case EFailed:
    100                 return false;
    101             }
    102 
    103             return false;
    104         }
    105 
    106     private:
    107         std::string _strCommandName;
    108         RemoteCommandParser _pfnParser;
    109         size_t _minArgumentCount;
    110         std::string _strHelp;
    111         std::string _strDescription;
    112     };
    113 
    114 public:
    115     TRemoteCommandHandlerTemplate(CCommandParser *pCommandParser)
    116         : _pCommandParser(pCommandParser), _maxCommandUsageLength(0)
    117     {
    118         // Help Command
    119         addCommandParser("help", nullptr, 0, "", "Show commands description and usage");
    120     }
    121     ~TRemoteCommandHandlerTemplate() override
    122     {
    123         // FIXME use unique_ptr
    124         for (auto *parser : _remoteCommandParserVector) {
    125 
    126             delete parser;
    127         }
    128     }
    129 
    130     // Parsers
    131     bool addCommandParser(const std::string &strCommandName, RemoteCommandParser pfnParser,
    132                           size_t minArgumentCount, const std::string &strHelp,
    133                           const std::string &strDescription)
    134     {
    135         if (findCommandParserItem(strCommandName)) {
    136 
    137             // Already exists
    138             return false;
    139         }
    140 
    141         // Add command
    142         _remoteCommandParserVector.push_back(new CRemoteCommandParserItem(
    143             strCommandName, pfnParser, minArgumentCount, strHelp, strDescription));
    144 
    145         return true;
    146     }
    147 
    148 private:
    149     // Command processing
    150     bool remoteCommandProcess(const IRemoteCommand &remoteCommand, std::string &strResult) override
    151     {
    152         // Dispatch
    153         const CRemoteCommandParserItem *pRemoteCommandParserItem =
    154             findCommandParserItem(remoteCommand.getCommand());
    155 
    156         if (!pRemoteCommandParserItem) {
    157 
    158             // Not found
    159             strResult = "Command not found!\nUse \"help\" to show available commands";
    160 
    161             return false;
    162         }
    163 
    164         if (remoteCommand.getCommand() == "help") {
    165 
    166             helpCommandProcess(strResult);
    167 
    168             return true;
    169         }
    170 
    171         return pRemoteCommandParserItem->parse(_pCommandParser, remoteCommand, strResult);
    172     }
    173 
    174     // Max command usage length, use for formatting
    175     void initMaxCommandUsageLength()
    176     {
    177         if (!_maxCommandUsageLength) {
    178             // Show usages
    179             for (const auto *pRemoteCommandParserItem : _remoteCommandParserVector) {
    180 
    181                 size_t remoteCommandUsageLength = pRemoteCommandParserItem->usage().length();
    182 
    183                 if (remoteCommandUsageLength > _maxCommandUsageLength) {
    184 
    185                     _maxCommandUsageLength = remoteCommandUsageLength;
    186                 }
    187             }
    188         }
    189     }
    190 
    191     /////////////////// Remote command parsers
    192     /// Help
    193     void helpCommandProcess(std::string &strResult)
    194     {
    195         initMaxCommandUsageLength();
    196 
    197         // Show usages
    198         for (const auto *pRemoteCommandParserItem : _remoteCommandParserVector) {
    199 
    200             std::string strUsage = pRemoteCommandParserItem->usage();
    201 
    202             // Align
    203             size_t spacesToAdd = _maxCommandUsageLength + 5 - strUsage.length();
    204 
    205             strResult += strUsage + std::string(spacesToAdd, ' ') + "=> " +
    206                          pRemoteCommandParserItem->getDescription() + '\n';
    207         }
    208     }
    209 
    210     const CRemoteCommandParserItem *findCommandParserItem(const std::string &strCommandName) const
    211     {
    212         for (const auto *pRemoteCommandParserItem : _remoteCommandParserVector) {
    213 
    214             if (pRemoteCommandParserItem->getCommandName() == strCommandName) {
    215 
    216                 return pRemoteCommandParserItem;
    217             }
    218         }
    219         return nullptr;
    220     }
    221 
    222 private:
    223     CCommandParser *_pCommandParser;
    224     std::vector<CRemoteCommandParserItem *> _remoteCommandParserVector;
    225     size_t _maxCommandUsageLength;
    226 };
    227