GM6000 Digital Heater Controller Branch: main
SDX-1330
Processor.h
Go to the documentation of this file.
1#ifndef Cpl_TShell_Processor_h_
2#define Cpl_TShell_Processor_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file */
14
15#include "colony_config.h"
16#include "Cpl/TShell/Context_.h"
17#include "Cpl/System/Mutex.h"
18#include "Cpl/System/Api.h"
19#include "Cpl/Text/FString.h"
22
23
24/** This symbol defines the size, in bytes, of the maximum allowed input
25 string/command.
26 */
27#ifndef OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE
28#define OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE 128
29#endif
30
31 /** This symbol defines the size, in bytes, of the maximum allowed unframed
32 output string/command.
33 */
34#ifndef OPTION_CPL_TSHELL_PROCESSOR_OUTPUT_SIZE
35#define OPTION_CPL_TSHELL_PROCESSOR_OUTPUT_SIZE 256
36#endif
37
38/** This symbols defines the Shell's greeting message
39 */
40#ifndef OPTION_CPL_TSHELL_PROCESSOR_GREETING
41#define OPTION_CPL_TSHELL_PROCESSOR_GREETING "\n--- Your friendly neighborhood TShell. ---\n\n\n"
42#endif
43
44/** This symbols defines the Shell's farewell message
45 */
46#ifndef OPTION_CPL_TSHELL_PROCESSOR_FAREWELL
47#define OPTION_CPL_TSHELL_PROCESSOR_FAREWELL "\n--- ...I am melting, am melting... ---\n\n"
48#endif
49
50/** This symbols defines the Shell's prompt string
51 */
52#ifndef OPTION_CPL_TSHELL_PROCESSOR_PROMPT
53#define OPTION_CPL_TSHELL_PROCESSOR_PROMPT "$ "
54#endif
55
56///
57namespace Cpl {
58///
59namespace TShell {
60
61/** This concrete class provides the implementation of Command Processor for
62 a TShell engine.
63
64 The implementation assumes a single threaded model, i.e. the Command
65 Processor and all of its commands run in a single thread. It is
66 APPLICATION's responsibility to provide any desired multi-threaded
67 support. There are two caveats to the single-threaded model:
68
69 o The output of the commands are mutex protected. This allows the
70 Output stream to be 'shared' with other sub-systems and/or
71 threads (e.g. the shell shares the same Output stream as the
72 Cpl::System::Trace logging output).
73
74 o The stop() method can be called safely from other threads.
75
76
77 Commands have the following syntax:
78
79 o A command starts with a printable ASCII character and ends with
80 a newline.
81
82 o Non-printable ASCII characters are not allowed.
83
84 o Any line that starts with a '#' is treated as comment line and
85 is ignored.
86
87 o Format of a command is: verb [arg]* where the verb and arguments
88 are separated by spaces. Arguments can contain spaces character
89 by enclosing the argument with double quote characters. A double
90 quote character can be embedded inside a quoted string by preceding
91 it the double quote character with the escape character. The escape
92 character can be embedded by escaping the escape character.
93
94 The USE_CPL_TSHELL_PROCESSOR_SILENT_WHEN_PUBLIC switch is use
95 to alter the TShell's output behavior - to not 'say anything'
96 until a user has successfully been authenticated (using the 'User'
97 command
98
99 HOW TO Enable Security:
100
101 - Set OPTION_TSHELL_CMD_COMMAND_DEFAULT_PERMISSION_LEVEL to something
102 other than ePUBLIC. This will be the permission for all legacy (i.e.
103 not Security aware) TShell commands
104
105 - Recommend turning on the switch: USE_CPL_TSHELL_PROCESSOR_SILENT_WHEN_PUBLIC
106
107 - Include the Cpl::TShell::Cmd::User command and provide an
108 implementation of the Cpl::TShell::Security interface (that is passed
109 to the command's constructor).
110
111 - Optionally include new security aware commands
112 */
113class Processor : public Context_
114{
115public:
116 /** Constructor.
117
118 @param commands Set of supported commands
119 @param deframer Frame decoder used to identify individual command
120 strings within the raw Input stream
121 @param framer Frame encoder used to encapsulate the output of
122 command in the Output stream.
123 @param outputLock Mutex to be used for ensuring the atomic output
124 of the commands.
125 @param commentChar The comment character used to indicate that the
126 input string is a comment and should not be
127 executed.
128 @param argEscape Escape character to be used when escaping double
129 quote characters inside a quoted argument.
130 @param argDelimiter The delimiter character used to separate the
131 command verb and arguments.
132 @param argQuote The quote character used to 'double quote' a
133 argument string.
134 @param argTerminator The command terminator character.
135 @param initialPermissionLevel The initial minimum permission level that a user needs to issue command(s)
136 */
140 Cpl::System::Mutex& outputLock,
141 char commentChar='#',
142 char argEscape='`',
143 char argDelimiter=' ',
144 char argQuote='"',
145 char argTerminator='\n',
146 Security::Permission_T initialPermissionLevel = Security::ePUBLIC
147 );
148
149
150public:
151 /// See Cpl::TShell::ProcessorApi
152 bool start( Cpl::Io::Input& infd, Cpl::Io::Output& outfd, bool blocking=true ) noexcept;
153
154 /// See Cpl::TShell::ProcessorApi
155 int poll() noexcept;
156
157 /// See Cpl::TShell::ProcessorApi
158 void requestStop() noexcept;
159
160 /// See Cpl::TShell::ProcessorApi
161 char getEscapeChar() noexcept;
162
163 /// See Cpl::TShell::ProcessorApi
164 char getDelimiterChar() noexcept;
165
166 /// See Cpl::TShell::ProcessorApi
167 char getQuoteChar() noexcept;
168
169 /// See Cpl::TShell::ProcessorApi
170 char getTerminatorChar() noexcept;
171
172public:
173 /// See Cpl::TShell::Context_
174 Cpl::Container::Map<Command>& getCommands() noexcept;
175
176 /// See Cpl::TShell::Context_
177 bool writeFrame( const char* text ) noexcept;
178
179 /// See Cpl::TShell::Context_
180 bool writeFrame( const char* text, size_t maxBytes ) noexcept;
181
182 /// See Cpl::TShell::Context_
183 Cpl::Text::String& getOutputBuffer() noexcept;
184
185 /// See Cpl::TShell::Context_
186 Cpl::Text::String& getTokenBuffer() noexcept;
187
188 /// See Cpl::TShell::Context_
189 Cpl::Text::String& getTokenBuffer2() noexcept;
190
191 /// See Cpl::TShell::Context_
192 bool oobRead( void* buffer, int numBytes, int& bytesRead ) noexcept;
193
194 /// See Cpl::TShell::Context_
195 Security::Permission_T getUserPermissionLevel() const noexcept;
196
197 /// See Cpl::TShell::Context_
198 Security::Permission_T setUserPermissionLevel( Security::Permission_T newPermissionLevel ) noexcept;
199
200protected:
201 /** Helper method that attempts to execute the content of the de-framed/decoded
202 'inputString'. The method returns the result code of the execute
203 command. If 'inputString' is not a valid command, then the appropriate
204 error/result code is returned.
205 */
206 virtual Command::Result_T executeCommand( char* deframedInput, Cpl::Io::Output& outfd ) noexcept;
207
208 /// Helper method
209 bool outputCommandError( Command::Result_T result, const char* deframedInput ) noexcept;
210
211 /** Helper method that performs a 'single' read cycle of the input stream.
212 Returns 0 if successful. Returns 1 if exiting. Returns -1 on error
213 */
214 int getAndProcessFrame( Cpl::Io::Output& outfd ) noexcept;
215
216 /** Helper method that executes the decoder, i.e. logic to parse the incoming
217 text. Returns 1 if a full/valid frame was found. Returns 0 if input frame
218 is incomplete. Return -1 if an error occurred.
219 */
220 virtual int readInput( size_t& frameSize ) noexcept;
221
222protected:
223 /// Command list
224 Cpl::Container::Map<Command>& m_commands;
225
226 /// Raw input de-framer
227 Cpl::Text::Frame::StreamDecoder& m_deframer;
228
229 /// Output framer handle
230 Cpl::Text::Frame::StreamEncoder& m_framer;
231
232 /// Output lock
233 Cpl::System::Mutex& m_outLock;
234
235 /// User's permission level
237
238 /// Comment character
240
241 /// Argument Escape character
242 char m_esc;
243
244 /// Argument delimiter
245 char m_del;
246
247 /// Argument quote character
249
250 /// Argument terminator character
251 char m_term;
252
253 /// Set to true when 'command prompt' should be outputted
255
256 /// Current frame size
258
259 /// My run state
261
262 /// Input Frame buffer
264
265 /// Buffer that is used to construct output messages
267
268 /// Shared token work buffer
270
271 /// Shared token work buffer
273};
274
275
276}; // end namespaces
277};
278#endif // end header latch
#define OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE
This symbol defines the size, in bytes, of the maximum allowed input string/command.
Definition Processor.h:28
#define OPTION_CPL_TSHELL_PROCESSOR_OUTPUT_SIZE
This symbol defines the size, in bytes, of the maximum allowed unframed output string/command.
Definition Processor.h:35
This template class implements a THREAD SAFE Ring Buffer.
Definition RingBufferMT.h:33
This partially abstract class defines a interface for operating on an input stream (example of a stre...
Definition Input.h:37
This partially abstract class defines a interface for operating on an output stream (example of a str...
Definition Output.h:34
This mutex class defines the interface for a mutex that has "recursive" semantics.
Definition Mutex.h:33
This class defines the interface for a TShell command.
Definition Command.h:32
This Private Namespace class defines a "Context" for a TShell command.
Definition Context_.h:32
This concrete class provides the implementation of Command Processor for a TShell engine.
Definition Processor.h:114
Cpl::Text::String & getOutputBuffer() noexcept
See Cpl::TShell::Context_.
Cpl::Text::FString< OPTION_CPL_TSHELL_PROCESSOR_OUTPUT_SIZE > m_outputBuffer
Buffer that is used to construct output messages.
Definition Processor.h:266
virtual Command::Result_T executeCommand(char *deframedInput, Cpl::Io::Output &outfd) noexcept
Helper method that attempts to execute the content of the de-framed/decoded 'inputString'.
bool oobRead(void *buffer, int numBytes, int &bytesRead) noexcept
See Cpl::TShell::Context_.
Cpl::Container::Map< Command > & m_commands
Command list.
Definition Processor.h:224
Security::Permission_T getUserPermissionLevel() const noexcept
See Cpl::TShell::Context_.
char m_term
Argument terminator character.
Definition Processor.h:251
bool m_writeCommandPrompt
Set to true when 'command prompt' should be outputted.
Definition Processor.h:254
Cpl::Container::Map< Command > & getCommands() noexcept
See Cpl::TShell::Context_.
Cpl::Text::Frame::StreamEncoder & m_framer
Output framer handle.
Definition Processor.h:230
int poll() noexcept
See Cpl::TShell::ProcessorApi.
char getQuoteChar() noexcept
See Cpl::TShell::ProcessorApi.
virtual int readInput(size_t &frameSize) noexcept
Helper method that executes the decoder, i.e.
Cpl::Text::String & getTokenBuffer2() noexcept
See Cpl::TShell::Context_.
char m_inputBuffer[OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE+1]
Input Frame buffer.
Definition Processor.h:263
char m_comment
Comment character.
Definition Processor.h:239
char getEscapeChar() noexcept
See Cpl::TShell::ProcessorApi.
Security::Permission_T m_userPermLevel
User's permission level.
Definition Processor.h:236
char m_quote
Argument quote character.
Definition Processor.h:248
size_t m_frameSize
Current frame size.
Definition Processor.h:257
Cpl::Text::FString< OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE > m_tokenBuffer2
Shared token work buffer.
Definition Processor.h:272
Cpl::Text::Frame::StreamDecoder & m_deframer
Raw input de-framer.
Definition Processor.h:227
Processor(Cpl::Container::Map< Command > &commands, Cpl::Text::Frame::StreamDecoder &deframer, Cpl::Text::Frame::StreamEncoder &framer, Cpl::System::Mutex &outputLock, char commentChar='#', char argEscape='`', char argDelimiter=' ', char argQuote='"', char argTerminator='\n', Security::Permission_T initialPermissionLevel = Security::ePUBLIC )
Constructor.
int getAndProcessFrame(Cpl::Io::Output &outfd) noexcept
Helper method that performs a 'single' read cycle of the input stream.
bool start(Cpl::Io::Input &infd, Cpl::Io::Output &outfd, bool blocking=true) noexcept
See Cpl::TShell::ProcessorApi.
Cpl::Text::String & getTokenBuffer() noexcept
See Cpl::TShell::Context_.
char getTerminatorChar() noexcept
See Cpl::TShell::ProcessorApi.
bool writeFrame(const char *text) noexcept
See Cpl::TShell::Context_.
Cpl::Text::FString< OPTION_CPL_TSHELL_PROCESSOR_INPUT_SIZE > m_tokenBuffer
Shared token work buffer.
Definition Processor.h:269
bool outputCommandError(Command::Result_T result, const char *deframedInput) noexcept
Helper method.
char m_esc
Argument Escape character.
Definition Processor.h:242
Security::Permission_T setUserPermissionLevel(Security::Permission_T newPermissionLevel) noexcept
See Cpl::TShell::Context_.
char m_del
Argument delimiter.
Definition Processor.h:245
Cpl::System::Mutex & m_outLock
Output lock.
Definition Processor.h:233
char getDelimiterChar() noexcept
See Cpl::TShell::ProcessorApi.
void requestStop() noexcept
See Cpl::TShell::ProcessorApi.
bool m_running
My run state.
Definition Processor.h:260
This abstract defines the interface for validating a 'user login' for TShell.
Definition Security.h:31
Permission_T
Permissions levels.
Definition Security.h:34
@ ePUBLIC
No permissions, i.e.
Definition Security.h:35
This partially concrete class defines an interface a Text "Decoder" that has a Cpl::Io::Input stream ...
Definition StreamDecoder.h:34
This concrete class implements the Encoder API where the Output destination is a Cpl::Io::Output stre...
Definition StreamEncoder.h:36
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20