mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
processInterface.cpp
Go to the documentation of this file.
1/** \file processInterface.cpp
2 * \author Jared R. Males (jaredmales@gmail.com)
3 * \brief Process interface facilities
4 * \ingroup IPC
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2021 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
28#include "mxError.hpp"
29
30#include <cstring>
31#include <sstream>
32#include <cstdio>
33#include <iostream>
34
35#include <unistd.h>
36#include <sys/wait.h>
37
38namespace mx
39{
40namespace ipc
41{
42
43int command_response( const char *cmd, char *resp, size_t respsz )
44{
45 FILE *cmd_ptr;
46
47 if( ( cmd_ptr = popen( cmd, "r" ) ) != NULL )
48 {
49 size_t written = 0;
50
51 char buf[MX_IPC_PI_BUFSZ];
52
53 while( fgets( buf, MX_IPC_PI_BUFSZ, cmd_ptr ) != NULL && written + 1 < respsz )
54 {
55 strcat( resp, buf );
56 written = strlen( resp );
57 }
58 pclose( cmd_ptr );
59 }
60 else
61 {
62 return -1;
63 }
64
65 return 0;
66}
67
69 int &retVal, // [out] the return value of the process. Only meaningful if this returns 0.
70 std::vector<std::string>
71 &commandOutput, // [out] the output, line by line. If an error, first entry contains the message.
72 std::vector<std::string> &commandStderr, // [out] the output of stderr.
73 const std::vector<std::string> &commandList // [in] command to be run, with one entry per command line word
74)
75{
76 int link[2];
77 int errlink[2];
78
79 pid_t pid;
80
81 if( pipe( link ) == -1 )
82 {
83 mxPError( "mx::ipc::runCommand", errno, "piping stdout" );
84 return -1;
85 }
86
87 if( pipe( errlink ) == -1 )
88 {
89 mxPError( "mx::ipc::runCommand", errno, "piping stderr" );
90 return -1;
91 }
92
93 if( ( pid = fork() ) == -1 )
94 {
95 mxPError( "mx::ipc::runCommand", errno, "forking" );
96 return -1;
97 }
98
99 if( pid == 0 )
100 {
101 if( dup2( link[1], STDOUT_FILENO ) < 0 )
102 {
103 mxPError( "mx::ipc::runCommand", errno, "dup2" );
104 return -1;
105 }
106
107 if( close( link[0] ) < 0 )
108 {
109 mxPError( "mx::ipc::runCommand", errno, "close" );
110 return -1;
111 }
112
113 if( close( link[1] ) < 0 )
114 {
115 mxPError( "mx::ipc::runCommand", errno, "close" );
116 return -1;
117 }
118
119 if( dup2( errlink[1], STDERR_FILENO ) < 0 )
120 {
121 mxPError( "mx::ipc::runCommand", errno, "dup2" );
122 return -1;
123 }
124
125 if( close( errlink[0] ) < 0 )
126 {
127 mxPError( "mx::ipc::runCommand", errno, "close" );
128 return -1;
129 }
130
131 if( close( errlink[1] ) < 0 )
132 {
133 mxPError( "mx::ipc::runCommand", errno, "close" );
134 return -1;
135 }
136
137 std::vector<const char *> charCommandList( commandList.size() + 1, NULL );
138
139 for( int index = 0; index < (int)commandList.size(); ++index )
140 {
141 charCommandList[index] = commandList[index].c_str();
142 }
143 execvp( charCommandList[0], const_cast<char **>( charCommandList.data() ) );
144
145 mxPError( "mx::ipc::runCommand", errno, "execvp returned" );
146
147 return -1;
148 }
149 else
150 {
151 char commandOutput_c[4096];
152
153 int status;
154
155 pid_t rvid = waitpid( pid, &status, 0 );
156
157 if( rvid < 0 )
158 {
159 mxPError( "mx::ipc::runCommand", errno, "waitpid" );
160 return -1;
161 }
162
163 if( WIFEXITED( status ) )
164 {
165 retVal = WEXITSTATUS( status );
166 }
167 else
168 {
169 mxError( "mx::ipc::runCommand", MXE_PROCERR, "child did not exit" );
170 return -1;
171 }
172
173 if( close( link[1] ) < 0 )
174 {
175 mxPError( "mx::ipc::runCommand", errno, "close" );
176 return -1;
177 }
178
179 if( close( errlink[1] ) < 0 )
180 {
181 mxPError( "mx::ipc::runCommand", errno, "close" );
182 return -1;
183 }
184
185 int rd;
186 if( ( rd = read( link[0], commandOutput_c, sizeof( commandOutput_c ) ) ) < 0 )
187 {
188 mxPError( "mx::ipc::runCommand", errno, "read" );
189
190 if( close( errlink[0] ) < 0 )
191 mxPError( "mx::ipc::runCommand", errno, "close" );
192 if( close( link[0] ) < 0 )
193 mxPError( "mx::ipc::runCommand", errno, "close" );
194
195 return -1;
196 }
197
198 if( close( link[0] ) < 0 )
199 {
200 mxPError( "mx::ipc::runCommand", errno, "close" );
201 if( close( errlink[0] ) < 0 )
202 mxPError( "mx::ipc::runCommand", errno, "close" );
203 return -1;
204 }
205
206 std::string line;
207
208 commandOutput_c[rd] = '\0';
209 std::string commandOutputString( commandOutput_c );
210
211 std::istringstream iss( commandOutputString );
212
213 while( getline( iss, line ) )
214 {
215 commandOutput.push_back( line );
216 }
217
218 //----stderr
219 if( ( rd = read( errlink[0], commandOutput_c, sizeof( commandOutput_c ) ) ) < 0 )
220 {
221 commandStderr.push_back( std::string( "Read error on stderr: " ) + strerror( errno ) );
222 if( close( errlink[0] ) < 0 )
223 mxPError( "mx::ipc::runCommand", errno, "close" );
224 return -1;
225 }
226
227 if( close( errlink[0] ) < 0 )
228 {
229 mxPError( "mx::ipc::runCommand", errno, "close" );
230 return -1;
231 }
232
233 commandOutput_c[rd] = '\0';
234 commandOutputString = commandOutput_c;
235
236 std::istringstream iss2( commandOutputString );
237
238 while( getline( iss2, line ) )
239 {
240 commandStderr.push_back( line );
241 }
242
243 return 0;
244 }
245}
246
247} // namespace ipc
248} // namespace mx
int command_response(const char *cmd, char *resp, size_t respsz)
Run a process and copy the output to a string.
int runCommand(int &retVal, std::vector< std::string > &commandOutput, std::vector< std::string > &commandStderr, const std::vector< std::string > &commandList)
Runs a command (with parameters) passed in using fork/exec.
Declares and defines the mxlib error reporting system.
The mxlib c++ namespace.
Definition mxError.hpp:106
Process interface facilities.